001 /* 002 * Copyright 2004 Niclas Hedhman 003 * Copyright 2004-2005 Stephen McConnell 004 * 005 * Licensed under the Apache License, Version 2.0 (the "License"); 006 * you may not use this file except in compliance with the License. 007 * You may obtain a copy of the License at 008 * 009 * http://www.apache.org/licenses/LICENSE-2.0 010 * 011 * Unless required by applicable law or agreed to in writing, software 012 * distributed under the License is distributed on an "AS IS" BASIS, 013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or 014 * implied. 015 * 016 * See the License for the specific language governing permissions and 017 * limitations under the License. 018 */ 019 020 package net.dpml.tools.tasks; 021 022 import java.io.File; 023 import java.io.FileFilter; 024 import java.io.FileInputStream; 025 import java.io.IOException; 026 import java.text.SimpleDateFormat; 027 import java.util.Calendar; 028 import java.util.Hashtable; 029 import java.util.Iterator; 030 import java.util.Map; 031 import java.util.Date; 032 import java.util.regex.Matcher; 033 import java.util.regex.Pattern; 034 035 import javax.xml.transform.Transformer; 036 import javax.xml.transform.TransformerFactory; 037 import javax.xml.transform.stream.StreamResult; 038 import javax.xml.transform.stream.StreamSource; 039 040 import net.dpml.transit.Transit; 041 042 import org.apache.tools.ant.BuildException; 043 import org.apache.tools.ant.Project; 044 import org.apache.tools.ant.taskdefs.Copy; 045 import org.apache.tools.ant.types.FileSet; 046 047 /** 048 * Site generation. 049 * 050 * @author <a href="http://www.dpml.net">The Digital Product Meta Library</a> 051 * @version 1.0.0 052 */ 053 public class DocTask extends GenericTask 054 { 055 private static final String ORG_NAME_VALUE = "The Digital Product Meta Library"; 056 private static final String DOC_TEMP_VALUE = "docs"; 057 private static final String DOC_SRC_VALUE = "docs"; 058 private static final String DOC_RESOURCES_VALUE = "resources"; 059 private static final String DOC_THEME_VALUE = "formal"; 060 private static final String DOC_FORMAT_VALUE = "html"; 061 private static final String DOC_DATE_FORMAT_VALUE = "yyyy-MMM-dd"; 062 private static final String DOC_STYLE_VALUE = "standard"; 063 private static final String DOC_ENTRY_VALUE = ""; 064 private static final String DOC_LOGO_RIGHT_FILE_VALUE = ""; 065 private static final String DOC_LOGO_RIGHT_URL_VALUE = ""; 066 private static final String DOC_LOGO_LEFT_FILE_VALUE = ""; 067 private static final String DOC_LOGO_LEFT_URL_VALUE = ""; 068 private static final String DOC_LOGO_MIDDLE_FILE_VALUE = ""; 069 private static final String DOC_LOGO_MIDDLE_URL_VALUE = ""; 070 private static final String DOC_BRAND_NAME_VALUE = "DPML"; 071 private static final String DOC_HOME_PATH_VALUE = "index.html"; 072 073 /** 074 * Constant organization key. 075 */ 076 public static final String ORG_NAME_KEY = "project.organization.name"; 077 078 /** 079 * Constant temp docs key. 080 */ 081 public static final String DOC_TEMP_KEY = "project.target.temp.docs"; 082 083 /** 084 * Constant docs src key. 085 */ 086 public static final String DOC_SRC_KEY = "project.docs.src"; 087 088 /** 089 * Constant docs resources key. 090 */ 091 public static final String DOC_RESOURCES_KEY = "project.docs.resources"; 092 093 /** 094 * Constant docs theme key. 095 */ 096 public static final String DOC_THEME_KEY = "project.docs.theme"; 097 098 /** 099 * Constant docs output format key. 100 */ 101 public static final String DOC_FORMAT_KEY = "project.docs.output.format"; 102 103 /** 104 * Constant docs date format key. 105 */ 106 public static final String DOC_DATE_FORMAT_KEY = "project.docs.date.format"; 107 108 /** 109 * Constant docs output style key. 110 */ 111 public static final String DOC_STYLE_KEY = "project.docs.output.style"; 112 113 /** 114 * Constant docs entry-point key. 115 */ 116 public static final String DOC_ENTRY_KEY = "project.docs.entry-point"; 117 118 /** 119 * Constant docs logo-right-file key. 120 */ 121 public static final String DOC_LOGO_RIGHT_FILE_KEY = "project.docs.logo.right.file"; 122 123 /** 124 * Constant docs logo-right url key. 125 */ 126 public static final String DOC_LOGO_RIGHT_URL_KEY = "project.docs.logo.right.url"; 127 128 /** 129 * Constant docs logo-left file key. 130 */ 131 public static final String DOC_LOGO_LEFT_FILE_KEY = "project.docs.logo.left.file"; 132 133 /** 134 * Constant docs logo-left url key. 135 */ 136 public static final String DOC_LOGO_LEFT_URL_KEY = "project.docs.logo.left.url"; 137 138 /** 139 * Constant docs logo-middle file key. 140 */ 141 public static final String DOC_LOGO_MIDDLE_FILE_KEY = "project.docs.logo.middle.file"; 142 143 /** 144 * Constant docs logo-middle url key. 145 */ 146 public static final String DOC_LOGO_MIDDLE_URL_KEY = "project.docs.logo.middle.url"; 147 148 /** 149 * Constant docs brand key. 150 */ 151 public static final String DOC_BRAND_NAME_KEY = "project.docs.brand.name"; 152 153 /** 154 * Constant docs anchor url key. 155 */ 156 public static final String DOC_ANCHOR_URL_KEY = "project.docs.anchor.url"; 157 158 /** 159 * Constant docs home page path. 160 */ 161 public static final String DOC_HOME_PATH_KEY = "project.docs.home.path"; 162 163 private String m_theme; 164 private File m_baseToDir; 165 private File m_baseSrcDir; 166 private File m_dest; 167 168 /** 169 * Return the assigned theme. 170 * @return the theme 171 */ 172 public String getTheme() 173 { 174 if( m_theme != null ) 175 { 176 return m_theme; 177 } 178 String theme = getProject().getProperty( DOC_THEME_KEY ); 179 if( null != theme ) 180 { 181 return theme; 182 } 183 else 184 { 185 return getResource().getProperty( DOC_THEME_KEY, "formal" ); 186 } 187 } 188 189 /** 190 * Set a directory for ultimate replication of the generated documentation. 191 * @param dir the utilimate destination directory 192 */ 193 public void setDest( File dir ) 194 { 195 m_dest = dir; 196 } 197 198 /** 199 * Set the doc theme. 200 * @param theme the theme name 201 */ 202 public void setTheme( final String theme ) 203 { 204 m_theme = theme; 205 } 206 207 /** 208 * Initialize the task. 209 * @exception BuildException if a build error occurs 210 */ 211 public void init() throws BuildException 212 { 213 if( !isInitialized() ) 214 { 215 super.init(); 216 final Project project = getProject(); 217 project.setNewProperty( ORG_NAME_KEY, ORG_NAME_VALUE ); 218 project.setNewProperty( DOC_SRC_KEY, DOC_SRC_VALUE ); 219 project.setNewProperty( DOC_RESOURCES_KEY, DOC_RESOURCES_VALUE ); 220 project.setNewProperty( DOC_FORMAT_KEY, DOC_FORMAT_VALUE ); 221 project.setNewProperty( DOC_DATE_FORMAT_KEY, DOC_DATE_FORMAT_VALUE ); 222 project.setNewProperty( DOC_STYLE_KEY, DOC_STYLE_VALUE ); 223 project.setNewProperty( DOC_ENTRY_KEY, DOC_ENTRY_VALUE ); 224 project.setNewProperty( DOC_TEMP_KEY, DOC_TEMP_VALUE ); 225 project.setNewProperty( DOC_LOGO_RIGHT_FILE_KEY, DOC_LOGO_RIGHT_FILE_VALUE ); 226 project.setNewProperty( DOC_LOGO_RIGHT_URL_KEY, DOC_LOGO_RIGHT_URL_VALUE ); 227 project.setNewProperty( DOC_LOGO_LEFT_FILE_KEY, DOC_LOGO_LEFT_FILE_VALUE ); 228 project.setNewProperty( DOC_LOGO_LEFT_URL_KEY, DOC_LOGO_LEFT_URL_VALUE ); 229 project.setNewProperty( DOC_LOGO_MIDDLE_FILE_KEY, DOC_LOGO_MIDDLE_FILE_VALUE ); 230 project.setNewProperty( DOC_LOGO_MIDDLE_URL_KEY, DOC_LOGO_MIDDLE_URL_VALUE ); 231 project.setNewProperty( DOC_BRAND_NAME_KEY, DOC_BRAND_NAME_VALUE ); 232 } 233 } 234 235 /** 236 * Execute the task. 237 */ 238 public void execute() 239 { 240 final Project project = getProject(); 241 final File srcDir = getContext().getTargetBuildDocsDirectory(); 242 if( !srcDir.exists() ) 243 { 244 return; 245 } 246 log( "Filtered source: " + srcDir.getAbsolutePath() ); 247 248 // 249 // create the temporary directory into which we generate the 250 // navigation structure (normally target/temp/docs) 251 // 252 253 final File temp = getContext().getTargetTempDirectory(); 254 final File destDir = new File( temp, "docs" ); 255 mkDir( destDir ); 256 257 // 258 // get the theme, output formats, etc. 259 // 260 261 final File docs = getContext().getTargetDocsDirectory(); 262 log( "Destination: " + docs.getAbsolutePath() ); 263 mkDir( docs ); 264 final String theme = getTheme(); 265 final String output = getOutputFormat(); 266 final String home = getHomePath(); 267 final File themeRoot = getThemesDirectory(); 268 final File themeDir = new File( themeRoot, theme + "/" + output ); 269 270 final File target = getContext().getTargetDirectory(); 271 final String resourcesPath = project.getProperty( DOC_RESOURCES_KEY ); 272 final File resources = new File( target, resourcesPath ); 273 274 log( "Year: " + getYear() ); 275 log( "Theme: " + themeDir ); 276 277 // 278 // initiate the transformation starting with the generation of 279 // the navigation structure based on the src directory content 280 // into the temporary destingation directory, copy the content 281 // sources to to the temp directory, transform the content and 282 // generated navigation in the temp dir using the selected them 283 // into the final docs directory, and copy over resources to 284 // the final docs directory 285 // 286 287 try 288 { 289 transformNavigation( themeDir, srcDir, destDir ); 290 copySources( srcDir, destDir ); 291 transformDocs( themeDir, destDir, docs ); 292 copyThemeResources( themeDir, docs ); 293 copySrcResources( resources, docs ); 294 } 295 catch( BuildException e ) 296 { 297 throw e; 298 } 299 catch( Throwable e ) 300 { 301 log( "XSLT execution failed: " + e.getMessage() ); 302 throw new BuildException( e ); 303 } 304 305 if( null != m_dest ) 306 { 307 copy( docs, m_dest, "**/*", "" ); 308 } 309 } 310 311 private File getThemesDirectory() 312 { 313 return new File( Transit.DPML_PREFS, "dpml/tools/themes" ); 314 } 315 316 private String getHomePath() 317 { 318 return getProject().getProperty( DOC_HOME_PATH_KEY ); 319 } 320 321 private String getOutputFormat() 322 { 323 return getProject().getProperty( DOC_FORMAT_KEY ); 324 } 325 326 private String getOutputStyle() 327 { 328 return getProject().getProperty( DOC_STYLE_KEY ); 329 } 330 331 private String getDateFormat() 332 { 333 return getProject().getProperty( DOC_DATE_FORMAT_KEY ); 334 } 335 336 private void transformNavigation( final File themeDir, final File source, final File dest ) 337 { 338 final File xslFile = new File( themeDir, "nav-aggregate.xsl" ); 339 if( !xslFile.exists() ) 340 { 341 return; // Theme may not use navigation. 342 } 343 log( "Transforming navigation." ); 344 try 345 { 346 transformTrax( 347 source, dest, xslFile, 348 "^.*/navigation.xml$", "", ".xml" ); 349 } 350 catch( BuildException e ) 351 { 352 throw e; 353 } 354 catch( Throwable e ) 355 { 356 final String error = 357 "Transformation failure: " + source; 358 throw new BuildException( error, e ); 359 } 360 } 361 362 private void copySources( final File source, final File dest ) 363 { 364 copy( source, dest, "**/*", "**/navigation.xml" ); 365 } 366 367 private void transformDocs( final File themeDir, final File build, final File docs ) 368 { 369 String style = getOutputStyle(); 370 File xslFile = new File( themeDir, style + ".xsl" ); 371 String output = getOutputFormat(); 372 log( "Transforming content." ); 373 transformTrax( 374 build, docs, xslFile, 375 "^.*\\.xml$", "^.*/navigation.xml$", "." + output ); 376 } 377 378 private void copySrcResources( final File resources, final File docs ) 379 { 380 copy( resources, docs, "**/*", "" ); 381 } 382 383 private void copyThemeResources( final File themeDir, final File docs ) 384 { 385 final File fromDir = new File( themeDir, "resources" ); 386 copy( fromDir, docs, "**/*", "" ); 387 } 388 389 private void copy( final File fromDir, final File toDir, final String includes, final String excludes ) 390 { 391 if( !fromDir.exists() ) 392 { 393 return; 394 } 395 396 final FileSet from = new FileSet(); 397 from.setDir( fromDir ); 398 from.setIncludes( includes ); 399 from.setExcludes( excludes ); 400 401 mkDir( toDir ); 402 403 final Copy copy = (Copy) getProject().createTask( "copy" ); 404 copy.setTodir( toDir ); 405 copy.addFileset( from ); 406 copy.setPreserveLastModified( true ); 407 copy.execute(); 408 } 409 410 411 private void transformTrax( 412 final File srcDir, final File toDir, final File xslFile, 413 final String includes, final String excludes, final String extension ) 414 throws BuildException 415 { 416 FileInputStream fis = null; 417 try 418 { 419 ClassLoader cl = getClass().getClassLoader(); 420 Thread.currentThread().setContextClassLoader( cl ); 421 StreamSource source = new StreamSource( xslFile ); 422 final TransformerFactory tfactory = TransformerFactory.newInstance(); 423 final Transformer transformer = tfactory.newTransformer( source ); 424 final RegexpFilter filter = new RegexpFilter( includes, excludes ); 425 426 m_baseToDir = toDir; 427 m_baseSrcDir = srcDir.getAbsoluteFile(); 428 String entrypoint = getEntryPoint(); 429 if( "".equals( entrypoint ) ) 430 { 431 transform( transformer, m_baseSrcDir, toDir, filter, extension ); 432 } 433 else 434 { 435 File fileToConvert = new File( m_baseSrcDir, entrypoint ); 436 transformFile( 437 transformer, fileToConvert, m_baseSrcDir, entrypoint, extension, m_baseToDir ); 438 } 439 } 440 catch( BuildException e ) 441 { 442 throw e; 443 } 444 catch( Exception e ) 445 { 446 throw new BuildException( e.getMessage(), e ); 447 } 448 finally 449 { 450 if( fis != null ) 451 { 452 try 453 { 454 fis.close(); 455 } 456 catch( IOException f ) 457 { 458 log( f.toString() ); 459 } 460 } 461 } 462 } 463 464 private void transform( final Transformer transformer, final File srcDir, final File toDir, 465 final FileFilter filter, final String extension ) 466 throws BuildException 467 { 468 boolean recursive = isRecursive(); 469 final File[] content = srcDir.listFiles( filter ); 470 for( int i=0; i < content.length; i++ ) 471 { 472 String base = content[i].getName(); 473 if( content[i].isDirectory() && recursive ) 474 { 475 final File newDest = new File( toDir, base ); 476 newDest.mkdirs(); 477 transform( transformer, content[i], newDest, filter, extension ); 478 } 479 if( content[i].isFile() ) 480 { 481 transformFile( transformer, content[i], srcDir, base, extension, toDir ); 482 } 483 } 484 } 485 486 private void transformFile( Transformer transformer, File content, File srcDir, 487 String base, String extension, File toDir ) 488 { 489 String userDir = System.getProperty( "user.dir" ); 490 System.setProperty( "user.dir", toDir.getAbsolutePath() ); 491 final String year = getYear(); 492 final String org = getOrganization(); 493 final String copyright = 494 "Copyright " 495 + year 496 + ", " 497 + org 498 + " All rights reserved."; 499 500 final String svnRoot = getProject().getProperty( DOC_ANCHOR_URL_KEY ); 501 final String svnSource = svnRoot + getRelSrcPath( srcDir ) + "/" + base; 502 503 final int pos = base.lastIndexOf( '.' ); 504 if( pos > 0 ) 505 { 506 base = base.substring( 0, pos ); 507 } 508 base = base + extension; 509 510 final File newDest = new File( toDir, base ); 511 final StreamSource xml = new StreamSource( content ); 512 final StreamResult out = new StreamResult( newDest ); 513 514 transformer.clearParameters(); 515 transformer.setParameter( "directory", getRelToPath( toDir ) ); 516 transformer.setParameter( "fullpath", getRelToPath( newDest ) ); 517 transformer.setParameter( "file", base ); 518 transformer.setParameter( "svn-location", svnSource ); 519 transformer.setParameter( "copyright", copyright ); 520 transformer.setParameter( 521 "logoright_file", 522 getProject().getProperty( DOC_LOGO_RIGHT_FILE_KEY ).trim() ); 523 transformer.setParameter( 524 "logoright_url", 525 getProject().getProperty( DOC_LOGO_RIGHT_URL_KEY ).trim() ); 526 transformer.setParameter( 527 "logoleft_file", 528 getProject().getProperty( DOC_LOGO_LEFT_FILE_KEY ).trim() ); 529 transformer.setParameter( 530 "logoleft_url", 531 getProject().getProperty( DOC_LOGO_LEFT_URL_KEY ).trim() ); 532 transformer.setParameter( 533 "logomiddle_file", 534 getProject().getProperty( DOC_LOGO_MIDDLE_FILE_KEY ).trim() ); 535 transformer.setParameter( 536 "logomiddle_url", 537 getProject().getProperty( DOC_LOGO_MIDDLE_URL_KEY ).trim() ); 538 transformer.setParameter( 539 "brand_name", 540 getProject().getProperty( DOC_BRAND_NAME_KEY ).trim() ); 541 transformer.setParameter( "generated_date", getNow() ); 542 setOtherProperties( transformer ); 543 try 544 { 545 transformer.transform( xml, out ); 546 } 547 catch( BuildException e ) 548 { 549 throw e; 550 } 551 catch( Throwable e ) 552 { 553 final String error = 554 "An error occured while attempting to transform document: " 555 + getRelToPath( newDest ); 556 throw new BuildException( error, e, getLocation() ); 557 } 558 finally 559 { 560 System.setProperty( "user.dir", userDir ); 561 } 562 } 563 564 private String getRelToPath( final File dir ) 565 { 566 final String basedir = m_baseToDir.getAbsolutePath(); 567 final String curdir = dir.getAbsolutePath(); 568 return curdir.substring( basedir.length() ); 569 } 570 571 private String getRelSrcPath( final File dir ) 572 { 573 final String basedir = m_baseSrcDir.getAbsolutePath(); 574 final String curdir = dir.getAbsolutePath(); 575 return curdir.substring( basedir.length() ); 576 } 577 578 /** 579 * Utility regualar expression filter. 580 */ 581 public class RegexpFilter implements FileFilter 582 { 583 private Pattern m_includes; 584 private Pattern m_excludes; 585 586 /** 587 * New filter creation. 588 * @param includes the includes 589 * @param excludes the excludes 590 */ 591 public RegexpFilter( final String includes, final String excludes ) 592 { 593 m_includes = Pattern.compile( includes ); 594 m_excludes = Pattern.compile( excludes ); 595 } 596 597 /** 598 * Test supplied file for acceptance. 599 * @param file the candidate 600 * @return TRUE if acceptable 601 */ 602 public boolean accept( final File file ) 603 { 604 final String basename = file.getName(); 605 606 if( basename.equals( ".svn" ) ) 607 { 608 return false; 609 } 610 611 if( basename.equals( "CVS" ) ) 612 { 613 return false; 614 } 615 616 if( file.isDirectory() ) 617 { 618 return true; 619 } 620 621 final String fullpath = file.getAbsolutePath().replace( '\\', '/' ); 622 623 Matcher m = m_includes.matcher( fullpath ); 624 if( !m.matches() ) 625 { 626 return false; 627 } 628 629 m = m_excludes.matcher( fullpath ); 630 return !m.matches(); 631 } 632 } 633 634 private String getYear() 635 { 636 String year = getProject().getProperty( "magic.year" ); 637 if( year != null ) 638 { 639 return year; 640 } 641 else 642 { 643 Calendar cal = Calendar.getInstance(); 644 return Integer.toString( cal.get( Calendar.YEAR ) ); 645 } 646 } 647 648 private String getOrganization() 649 { 650 return getProject().getProperty( ORG_NAME_KEY ); 651 } 652 653 private boolean isRecursive() 654 { 655 return "".equals( getEntryPoint() ); 656 } 657 658 private String getEntryPoint() 659 { 660 String point = getProject().getProperty( DOC_ENTRY_KEY ); 661 if( point == null ) 662 { 663 return ""; 664 } 665 return point; 666 } 667 668 private void setOtherProperties( Transformer transformer ) 669 { 670 String prefix = "project.docs.xsl."; 671 int prefixLen = prefix.length(); 672 673 Hashtable p = getProject().getProperties(); 674 Iterator list = p.entrySet().iterator(); 675 while( list.hasNext() ) 676 { 677 Map.Entry entry = (Map.Entry) list.next(); 678 String key = (String) entry.getKey(); 679 if( key.startsWith( prefix ) ) 680 { 681 String value = (String) entry.getValue(); 682 key = key.substring( prefixLen ); 683 transformer.setParameter( key, value ); 684 log( "Setting " + key + "=" + value ); 685 } 686 } 687 } 688 689 private String getNow() 690 { 691 Date now = new Date(); 692 SimpleDateFormat sdf = new SimpleDateFormat( getDateFormat() ); 693 String result = sdf.format( now ); 694 return result; 695 } 696 }